package com.lazy.book.common.utils;


import org.springframework.util.AntPathMatcher;

import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import static org.apache.commons.lang3.StringUtils.containsIgnoreCase;
import static org.apache.commons.lang3.StringUtils.isBlank;

/**
 *
 * @author lazy
 * @since 2025-09-04
 */
public class StringUtils {

  private StringUtils() {
    throw new IllegalStateException("Utility class");
  }

  /**
   * 空字符串
   */
  private static final String NULL_STR = "";

  /**
   * 下划线
   */
  private static final char SEPARATOR = '_';

  /**
   * 将字符串的首字母转换为小写。
   *
   * @param s
   *            要转换的字符串
   * @return 首字母小写的字符串，如果首字母已是小写，则返回原字符串
   * @author sz
   * @since 2021-09-15
   */
  public static String toLowerCaseFirstOne(String s) {
    if (Character.isLowerCase(s.charAt(0))) {
      return s;
    } else {
      return Character.toLowerCase(s.charAt(0)) + s.substring(1);
    }
  }

  /**
   * 将字符串的首字母转换为大写。
   *
   * @param s
   *            要转换的字符串
   * @return 首字母大写的字符串，如果首字母已是大写，则返回原字符串
   * @author sz
   * @since 2021-09-15
   */
  public static String toUpperCaseFirstOne(String s) {
    if (Character.isUpperCase(s.charAt(0))) {
      return s;
    } else {
      return Character.toUpperCase(s.charAt(0)) + s.substring(1);
    }
  }

  /**
   * 使用正则表达式替换字符串中的匹配部分。
   *
   * @param str
   *            源字符串
   * @param pattern
   *            正则表达式
   * @param replaceArrValue
   *            替换值的数组，按顺序应用
   * @return 替换后的字符串
   * @since 2021-11-25 14:19:50
   */
  public static String getRealKey(String str, String pattern, String[] replaceArrValue) {
    Matcher match = Pattern.compile(pattern).matcher(str);
    List<String> matchList = new ArrayList<>();
    while (match.find()) {
      matchList.add(match.group(1));
    }
    for (int i = 0; i < replaceArrValue.length; i++) {
      str = str.replace(matchList.get(i), replaceArrValue[i]);
    }
    return str;
  }

  public static String getRealKey(String sourceKey, String... replaceArrValue) {
    String pattern = "(\\$\\{\\w+\\})";
    return getRealKey(sourceKey, pattern, replaceArrValue);
  }

  public static String replacePlaceholders(String input, String... args) {
    // 编译一个正则表达式模式，用于匹配 ${...} 形式的占位符
    Pattern pattern = Pattern.compile("\\$\\{([^}]+)}");
    Matcher matcher = pattern.matcher(input);

    // 使用StringBuilder来构建最终的字符串，它比StringBuffer更高效
    StringBuilder result = new StringBuilder();

    // 用于跟踪args参数的索引
    int index = 0;

    // 使用Matcher的find和appendReplacement方法来逐步替换占位符
    while (matcher.find()) {
      // 检查是否有足够的参数来替换占位符
      String replacement = (index < args.length) ? args[index] : matcher.group();
      matcher.appendReplacement(result, Matcher.quoteReplacement(replacement));
      index++;
    }

    // 追加未替换部分的尾部到结果中
    matcher.appendTail(result);

    // 返回最终构建的字符串
    return result.toString();
  }

  public static String subWithLength(String str, int start, int length) {
    // 防止索引越界
    if (start < 0) {
      start = 0;
    }
    if (start >= str.length()) {
      return "";
    }

    // 计算截取的结束索引
    int end = Math.min(start + length, str.length());

    return str.substring(start, end);
  }

  public static String toSnakeCase(String camelCase) {
    if (camelCase == null || camelCase.isEmpty()) {
      return camelCase;
    }

    StringBuilder result = new StringBuilder();
    result.append(camelCase.substring(0, 1).toLowerCase()); // 将首字母转小写并添加到结果

    for (int i = 1; i < camelCase.length(); i++) {
      char c = camelCase.charAt(i);
      if (Character.isUpperCase(c)) {
        // 如果当前字符是大写字母，添加下划线并转换为小写
        result.append("_").append(Character.toLowerCase(c));
      } else {
        // 否则，直接添加当前字符
        result.append(c);
      }
    }
    return result.toString();
  }

  /**
   * 获取参数不为空值
   *
   * @param value
   *            defaultValue 要判断的value
   * @return value 返回值
   */
  public static <T> T nvl(T value, T defaultValue) {
    return value != null ? value : defaultValue;
  }

  /**
   * * 判断一个Collection是否为空， 包含List，Set，Queue
   *
   * @param coll
   *            要判断的Collection
   * @return true：为空 false：非空
   */
  public static boolean isEmpty(Collection<?> coll) {
    return isNull(coll) || coll.isEmpty();
  }

  /**
   * * 判断一个Collection是否非空，包含List，Set，Queue
   *
   * @param coll
   *            要判断的Collection
   * @return true：非空 false：空
   */
  public static boolean isNotEmpty(Collection<?> coll) {
    return !isEmpty(coll);
  }

  /**
   * * 判断一个对象数组是否为空
   *
   * @param objects
   *            要判断的对象数组 * @return true：为空 false：非空
   */
  public static boolean isEmpty(Object[] objects) {
    return isNull(objects) || (objects.length == 0);
  }

  /**
   * * 判断一个对象数组是否非空
   *
   * @param objects
   *            要判断的对象数组
   * @return true：非空 false：空
   */
  public static boolean isNotEmpty(Object[] objects) {
    return !isEmpty(objects);
  }

  /**
   * * 判断一个Map是否为空
   *
   * @param map
   *            要判断的Map
   * @return true：为空 false：非空
   */
  public static boolean isEmpty(Map<?, ?> map) {
    return isNull(map) || map.isEmpty();
  }

  /**
   * * 判断一个Map是否为空
   *
   * @param map
   *            要判断的Map
   * @return true：非空 false：空
   */
  public static boolean isNotEmpty(Map<?, ?> map) {
    return !isEmpty(map);
  }

  /**
   * * 判断一个字符串是否为空串
   *
   * @param str
   *            String
   * @return true：为空 false：非空
   */
  public static boolean isEmpty(String str) {
    return isNull(str) || NULL_STR.equals(str.trim());
  }

  /**
   * * 判断一个字符串是否为非空串
   *
   * @param str
   *            String
   * @return true：非空串 false：空串
   */
  public static boolean isNotEmpty(String str) {
    return !isEmpty(str);
  }

  /**
   * * 判断一个对象是否为空
   *
   * @param object
   *            Object
   * @return true：为空 false：非空
   */
  public static boolean isNull(Object object) {
    return object == null;
  }

  /**
   * * 判断一个对象是否非空
   *
   * @param object
   *            Object
   * @return true：非空 false：空
   */
  public static boolean isNotNull(Object object) {
    return !isNull(object);
  }

  /**
   * * 判断一个对象是否是数组类型（Java基本型别的数组）
   *
   * @param object
   *            对象
   * @return true：是数组 false：不是数组
   */
  public static boolean isArray(Object object) {
    return isNotNull(object) && object.getClass().isArray();
  }

  /**
   * 去空格
   */
  public static String trim(String str) {
    return (str == null ? "" : str.trim());
  }

  /**
   * 截取字符串
   *
   * @param str
   *            字符串
   * @param start
   *            开始
   * @return 结果
   */
  public static String substring(final String str, int start) {
    if (str == null) {
      return NULL_STR;
    }

    if (start < 0) {
      start = str.length() + start;
    }

    if (start < 0) {
      start = 0;
    }
    if (start > str.length()) {
      return NULL_STR;
    }

    return str.substring(start);
  }

  /**
   * 截取字符串
   *
   * @param str
   *            字符串
   * @param start
   *            开始
   * @param end
   *            结束
   * @return 结果
   */
  public static String substring(final String str, int start, int end) {
    if (str == null) {
      return NULL_STR;
    }

    if (end < 0) {
      end = str.length() + end;
    }
    if (start < 0) {
      start = str.length() + start;
    }

    if (end > str.length()) {
      end = str.length();
    }

    if (start > end) {
      return NULL_STR;
    }

    if (start < 0) {
      start = 0;
    }
    if (end < 0) {
      end = 0;
    }

    return str.substring(start, end);
  }

  /**
   * 字符串转set
   *
   * @param str
   *            字符串
   * @param sep
   *            分隔符
   * @return set集合
   */
  public static Set<String> str2Set(String str, String sep) {
    return new HashSet<>(str2List(str, sep, true, false));
  }

  /**
   * 字符串转list
   *
   * @param str
   *            字符串
   * @param sep
   *            分隔符
   * @param filterBlank
   *            过滤纯空白
   * @param trim
   *            去掉首尾空白
   * @return list集合
   */
  public static List<String> str2List(String str, String sep, boolean filterBlank, boolean trim) {
    List<String> list = new ArrayList<>();
    if (StringUtils.isEmpty(str)) {
      return list;
    }

    // 过滤空白字符串
    if (filterBlank && isBlank(str)) {
      return list;
    }
    String[] split = str.split(sep);
    for (String string : split) {
      if (filterBlank && isBlank(string)) {
        continue;
      }
      if (trim) {
        string = string.trim();
      }
      list.add(string);
    }

    return list;
  }

  /**
   * 判断给定的collection列表中是否包含数组array 判断给定的数组array中是否包含给定的元素value
   *
   * @param collection
   *            给定的集合
   * @param array
   *            给定的数组
   * @return boolean 结果
   */
  public static boolean containsAny(Collection<String> collection, String... array) {
    if (isEmpty(collection) || isEmpty(array)) {
      return false;
    } else {
      for (String str : array) {
        if (collection.contains(str)) {
          return true;
        }
      }
      return false;
    }
  }

  /**
   * 查找指定字符串是否包含指定字符串列表中的任意一个字符串同时串忽略大小写
   *
   * @param cs
   *            指定字符串
   * @param searchCharSequences
   *            需要检查的字符串数组
   * @return 是否包含任意一个字符串
   */
  public static boolean containsAnyIgnoreCase(CharSequence cs, CharSequence... searchCharSequences) {
    if (isEmpty((Collection<?>) cs) || isEmpty(searchCharSequences)) {
      return false;
    }
    for (CharSequence testStr : searchCharSequences) {
      if (containsIgnoreCase(cs, testStr)) {
        return true;
      }
    }
    return false;
  }

  /**
   * 驼峰转下划线命名
   */
  public static String toUnderScoreCase(String str) {
    if (str == null) {
      return null;
    }
    StringBuilder sb = new StringBuilder();
    // 前置字符是否大写
    boolean preCharIsUpperCase;
    // 当前字符是否大写
    boolean curreCharIsUpperCase;
    // 下一字符是否大写
    boolean nexteCharIsUpperCase = true;
    for (int i = 0; i < str.length(); i++) {
      char c = str.charAt(i);
      if (i > 0) {
        preCharIsUpperCase = Character.isUpperCase(str.charAt(i - 1));
      } else {
        preCharIsUpperCase = false;
      }

      curreCharIsUpperCase = Character.isUpperCase(c);

      if (i < (str.length() - 1)) {
        nexteCharIsUpperCase = Character.isUpperCase(str.charAt(i + 1));
      }

      if (preCharIsUpperCase && curreCharIsUpperCase && !nexteCharIsUpperCase) {
        sb.append(SEPARATOR);
      } else if ((i != 0 && !preCharIsUpperCase) && curreCharIsUpperCase) {
        sb.append(SEPARATOR);
      }
      sb.append(Character.toLowerCase(c));
    }

    return sb.toString();
  }

  /**
   * 是否包含字符串
   *
   * @param str
   *            验证字符串
   * @param strs
   *            字符串组
   * @return 包含返回true
   */
  public static boolean inStringIgnoreCase(String str, String... strs) {
    if (str != null && strs != null) {
      for (String s : strs) {
        if (str.equalsIgnoreCase(trim(s))) {
          return true;
        }
      }
    }
    return false;
  }

  /**
   * 将下划线大写方式命名的字符串转换为驼峰式。如果转换前的下划线大写方式命名的字符串为空，则返回空字符串。
   * 例如：HELLO_WORLD->HelloWorld
   *
   * @param name
   *            转换前的下划线大写方式命名的字符串
   * @return 转换后的驼峰式命名的字符串
   */
  public static String convertToCamelCase(String name) {
    StringBuilder result = new StringBuilder();
    // 快速检查
    if (name == null || name.isEmpty()) {
      // 没必要转换
      return "";
    } else if (!name.contains("_")) {
      // 不含下划线，仅将首字母大写
      return name.substring(0, 1).toUpperCase() + name.substring(1);
    }
    // 用下划线将原始字符串分割
    String[] camels = name.split("_");
    for (String camel : camels) {
      // 跳过原始字符串中开头、结尾的下换线或双重下划线
      if (camel.isEmpty()) {
        continue;
      }
      // 首字母大写
      result.append(camel.substring(0, 1).toUpperCase());
      result.append(camel.substring(1).toLowerCase());
    }
    return result.toString();
  }

  /**
   * 驼峰式命名法 例如：user_name->userName
   */
  public static String toCamelCase(String s) {
    if (s == null) {
      return null;
    }
    if (s.indexOf(SEPARATOR) == -1) {
      return s;
    }
    s = s.toLowerCase();
    StringBuilder sb = new StringBuilder(s.length());
    boolean upperCase = false;
    for (int i = 0; i < s.length(); i++) {
      char c = s.charAt(i);

      if (c == SEPARATOR) {
        upperCase = true;
      } else if (upperCase) {
        sb.append(Character.toUpperCase(c));
        upperCase = false;
      } else {
        sb.append(c);
      }
    }
    return sb.toString();
  }

  /**
   * 查找指定字符串是否匹配指定字符串列表中的任意一个字符串
   *
   * @param str
   *            指定字符串
   * @param strs
   *            需要检查的字符串数组
   * @return 是否匹配
   */
  public static boolean matches(String str, List<String> strs) {
    if (isEmpty(str) || isEmpty(strs)) {
      return false;
    }
    for (String pattern : strs) {
      if (isMatch(pattern, str)) {
        return true;
      }
    }
    return false;
  }

  /**
   * 判断url是否与规则配置: ? 表示单个字符; * 表示一层路径内的任意字符串，不可跨层级; ** 表示任意层路径;
   *
   * @param pattern
   *            匹配规则
   * @param url
   *            需要匹配的url
   * @return 是否匹配
   */
  public static boolean isMatch(String pattern, String url) {
    AntPathMatcher matcher = new AntPathMatcher();
    return matcher.match(pattern, url);
  }

  @SuppressWarnings("unchecked")
  public static <T> T cast(Object obj) {
    return (T) obj;
  }

  /**
   * 数字左边补齐0，使之达到指定长度。注意，如果数字转换为字符串后，长度大于size，则只保留 最后size个字符。
   *
   * @param num
   *            数字对象
   * @param size
   *            字符串指定长度
   * @return 返回数字的字符串格式，该字符串为指定长度。
   */
  public static String padl(final Number num, final int size) {
    return padl(num.toString(), size, '0');
  }

  /**
   * 字符串左补齐。如果原始字符串s长度大于size，则只保留最后size个字符。
   *
   * @param s
   *            原始字符串
   * @param size
   *            字符串指定长度
   * @param c
   *            用于补齐的字符
   * @return 返回指定长度的字符串，由原字符串左补齐或截取得到。
   */
  public static String padl(final String s, final int size, final char c) {
    final StringBuilder sb = new StringBuilder(size);
    if (s != null) {
      final int len = s.length();
      if (s.length() <= size) {
        sb.append(String.valueOf(c).repeat(size - len));
        sb.append(s);
      } else {
        return s.substring(len - size, len);
      }
    } else {
      sb.append(String.valueOf(c).repeat(Math.max(0, size)));
    }
    return sb.toString();
  }

  public static String extractAndLowercase(String input) {
    if (input == null || input.isEmpty()) {
      return "";
    }

    int underscoreIndex = input.indexOf('_');
    if (underscoreIndex != -1) {
      // If underscore exists, extract substring before the first underscore
      return input.substring(0, underscoreIndex).toLowerCase();
    } else {
      // If no underscore, use the entire string and convert to lowercase
      return input.toLowerCase();
    }
  }

}
